#include"../logs/mylog.h"
#include<unistd.h>
/*
    扩展一个以时间作为日志切换的滚动类型 的日志落地模块
    1.以时间作为文件滚动，实际上是以时间段进行滚动
    实现思想：以当前系统时间，取模时间段大小，可以得到当前时间段时第几个时间段
    每次取模判断与当前文件的时间段是否一致，不一致代表不在一个时间段
    time(nullptr)%gap;11
    time(nullptr)%60 ==10 当前是第n个60秒
*/
enum class TimeGap
{
    GAP_SECOND,
    GAP_MINUTE,
    GAP_HOUR,
    GAP_DAY
};
class RollByTimeSink : public mylog::LogSink
{
public:
    // 构造时传入文件名，打开文件，并将操作句柄管理起来
    RollByTimeSink(const std::string &basename, TimeGap gap_type)
        : _basename(basename)
    {
        switch (gap_type)
        {
        case TimeGap::GAP_SECOND:
            _gap_size = 1;
            break;
        case TimeGap::GAP_MINUTE:
            _gap_size = 60;
            break;
        case TimeGap::GAP_HOUR:
            _gap_size = 3600;
            break;
        case TimeGap::GAP_DAY:
            _gap_size = 3600 * 24;
            break;
        }
        _cur_gap = _gap_size == 1 ? mylog::util::Date::now() : mylog::util::Date::now() % _gap_size; // 获取当前是第几个时间段
        std::string filename = createNewFile();
        mylog::util::File::createDirectory(mylog::util::File::getPath(filename));
        _ofs.open(filename, std::ios::binary | std::ios::app);
        assert(_ofs.is_open());
    }
    // 将日志消息写入到指定文件，判断当前时间是否是当前文件的时间段，不是则切换文件
    void log(const char *data, size_t len) override
    {
        time_t cur = mylog::util::Date::now();
        if ((cur % _gap_size) != _cur_gap)
        {
            _ofs.close();
            std::string filename = createNewFile();
            _ofs.open(filename, std::ios::binary | std::ios::app);
            assert(_ofs.is_open());
        }
        _ofs.write(data, len);
        assert(_ofs.good());
    }

private:
    std::string createNewFile()
    {
        time_t t = mylog::util::Date::now();
        struct tm lt;
        localtime_r(&t, &lt);
        std::stringstream filename;
        filename << _basename;
        filename << lt.tm_year + 1900;
        filename << lt.tm_mon + 1;
        filename << lt.tm_mday;
        filename << lt.tm_hour;
        filename << lt.tm_min;
        filename << lt.tm_sec;
        filename << ".log";

        return filename.str();
    }

private:
    std::string _basename;
    std::ofstream _ofs;
    size_t _cur_gap;  // 当前是第几个时间段
    size_t _gap_size; // 时间段的大小
};



int main()
{

    std::unique_ptr<mylog::LoggerBuilder> builder(new mylog::GlobalLoggerBuilder());
    builder->buildLoggerLevel(mylog::LogLevel::value::WARN);
    builder->buildLoggerName("async_logger");
    //builder->buildEnableUnSafeAsync();
    builder->buildFormatter("[%c][%f:%l]%m%n");
    builder->buildLoggerType(mylog::LoggerType::LOGGER_ASYNC);
    builder->buildSink<RollByTimeSink>("./logfile/roll-async-by-time",TimeGap::GAP_SECOND);
    mylog::Logger::ptr logger = builder->build();

    size_t cur =mylog::util::Date::now();
    while(mylog::util::Date::now()<cur+5)
    {
        logger->fatal("%s","这是一条测试日志");
        usleep(1000);
    }
    return 0;
}